@ampproject/remapping
Remap sequential sourcemaps through transformations to point at the original source code
Remapping allows you to take the sourcemaps generated through transforming your code and "remap"
them to the original source locations. Think "my minified code, transformed with babel and bundled
with webpack", all pointing to the correct location in your original source code.
With remapping, none of your source code transformations need to be aware of the input's sourcemap,
they only need to generate an output sourcemap. This greatly simplifies building custom
transformations (think a find-and-replace).
Installation
npm install @ampproject/remapping
Usage
function remapping(
map: SourceMap | SourceMap[],
loader: (file: string, ctx: LoaderContext) => (SourceMap | null | undefined),
options?: { excludeContent: boolean, decodedMappings: boolean }
): SourceMap;
type LoaderContext = {
readonly importer: string;
readonly depth: number;
source: string;
content: string | null | undefined;
}
remapping
takes the final output sourcemap, and a loader
function. For every source file pointer
in the sourcemap, the loader
will be called with the resolved path. If the path itself represents
a transformed file (it has a sourcmap associated with it), then the loader
should return that
sourcemap. If not, the path will be treated as an original, untransformed source code.
const transformedMap = JSON.stringify({
file: 'transformed.js',
mappings: ';CAEE',
sources: ['helloworld.js'],
version: 3,
});
const minifiedTransformedMap = JSON.stringify({
file: 'transformed.min.js',
mappings: 'AACC',
names: [],
sources: ['transformed.js'],
version: 3,
});
const remapped = remapping(
minifiedTransformedMap,
(file, ctx) => {
if (file === 'transformed.js') {
console.assert(ctx.importer === '');
console.assert(ctx.depth === 1);
return transformedMap;
}
console.assert(file === 'helloworld.js');
console.assert(ctx.importer === 'transformed.js');
console.assert(ctx.depth === 2);
return null;
}
);
console.log(remapped);
In this example, loader
will be called twice:
"transformed.js"
, the first source file pointer in the minifiedTransformedMap
. We return the
associated sourcemap for it (its a transformed file, after all) so that sourcemap locations can
be traced through it into the source files it represents."helloworld.js"
, our original, unmodified source code. This file does not have a sourcemap, so
we return null
.
The remapped
sourcemap now points from transformed.min.js
into locations in helloworld.js
. If
you were to read the mappings
, it says "0th column of the first line output line points to the 1st
column of the 2nd line of the file helloworld.js
".
Multiple transformations of a file
As a convenience, if you have multiple single-source transformations of a file, you may pass an
array of sourcemap files in the order of most-recent transformation sourcemap first. Note that this
changes the importer
and depth
of each call to our loader. So our above example could have been
written as:
const remapped = remapping(
[minifiedTransformedMap, transformedMap],
() => null
);
console.log(remapped);
Advanced control of the loading graph
source
The source
property can overridden to any value to change the location of the current load. Eg,
for an original source file, it allows us to change the location to the original source regardless
of what the sourcemap source entry says. And for transformed files, it allows us to change the
relative resolving location for child sources of the loaded sourcemap.
const remapped = remapping(
minifiedTransformedMap,
(file, ctx) => {
if (file === 'transformed.js') {
ctx.source = 'src/transformed.js';
return transformedMap;
}
console.assert(file === 'src/helloworld.js');
ctx.source = 'src/nested/transformed.js';
return null;
}
);
console.log(remapped);
content
The content
property can be overridden when we encounter an original source file. Eg, this allows
you to manually provide the source content of the original file regardless of whether the
sourcesContent
field is present in the parent sourcemap. It can also be set to null
to remove
the source content.
const remapped = remapping(
minifiedTransformedMap,
(file, ctx) => {
if (file === 'transformed.js') {
return transformedMap;
}
console.assert(file === 'helloworld.js');
ctx.content = fs.readFileSync(file, 'utf8');
return null;
}
);
console.log(remapped);
Options
excludeContent
By default, excludeContent
is false
. Passing { excludeContent: true }
will exclude the
sourcesContent
field from the returned sourcemap. This is mainly useful when you want to reduce
the size out the sourcemap.
decodedMappings
By default, decodedMappings
is false
. Passing { decodedMappings: true }
will leave the
mappings
field in a decoded state instead of
encoding into a VLQ string.